观察者模式(又称为发布-订阅(Publish/Subscribe)模式,属于行为型模式的一种,它是将行为独立模块化,降低了行为和主体的耦合性。它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态变化时,会通知所有的观察者对象,使他们能够自动更新自己。
- Subject:抽象主题(抽象被观察者),抽象主题角色把所有观察者对象保存在一个集合里,每个主题都可以有任意数量的观察者,抽象主题提供一个接口,可以增加和删除观察者对象。
- ConcreteSubject:具体主题(具体被观察者),该角色将有关状态存入具体观察者对象,在具体主题的内部状态发生改变时,给所有注册过的观察者发送通知。
- Observer:抽象观察者,是观察者者的抽象类,它定义了一个更新接口,使得在得到主题更改通知时更新自己。
- ConcrereObserver:具体观察者,是实现抽象观察者定义的更新接口,以便在得到主题更改通知时更新自身的状态。
PHP 内置了
- SplSubject 抽象主题 Interface
- SplObserver 抽象观察者 Interface
接口约束
// 主题 被观察者
interface SplSubject {
public function attach(SplObserver $observer); //注册观察者到当前主题
public function detach(SplObserver $observer); //从当前主题删除观察者
public function notify(); //主题状态更新时通知所有的观察者做相应的处理
}
// 观察者
interface SplObserver {
public function update(SplSubject $subject); //注册观察者到当前主题
}
通过项目中的实际应用能更容易的去理解观察者模式
下面我们以用户为主题,邮件模块和短信模块为观察者
当用户注册成功时,邮件观察者或短信观察者则收到相应的通知,发送邮件和短信给用户
User 主题
<?php
/**
* 主题类(被观察者相当于一个主题,观察者订阅这个主题)
* 当我们注册用户成功的时候想发送 email 和 sms 通知用户注册成功
* 则 可以将 SendEmail 和 SendSms 作为观察者
* 注册到 User 的观察者中
* 当 User register 成功时 notify 给 observers
* 各 observe 通过约定的 update 接口进行相应的处理 发邮件或发短信
*/
class User implements SplSubject
{
public $name;
public $email;
public $mobile;
/**
* 当前主题下的观察者集合
* @var array
*/
private $observers = [];
/**
* 模拟注册
* @param [type] $name [description]
* @param [type] $email [description]
* @param [type] $mobile [description]
* @return [type] [description]
*/
public function register($name, $email, $mobile)
{
$this->name = $name;
$this->email = $email;
$this->mobile = $mobile;
//business handle and register success
$reg_result = true;
if ($reg_result) {
$this->notify(); // 注册成功 所有的观察者将会收到此主题的通知
return true;
}
return false;
}
/**
* 当前主题注册新的观察者
* @param SplObserver $observer [description]
* @return [type] [description]
*/
public function attach(SplObserver $observer)
{
return array_push($this->observers, $observer);
}
/**
* 当前主题删除已注册的观察者
* @param SplObserver $observer [description]
* @return [type] [description]
*/
public function detach(SplObserver $observer)
{
$key = array_search($observer, $this->observers, true);
if (false !== $key) {
unset($this->observers[$key]);
return true;
}
return false;
}
/**
* 状态更新 通知所有的观察者
* @return [type] [description]
*/
public function notify()
{
if (! empty($this->observers)) {
foreach ($this->observers as $key => $observer) {
$observer->update($this);
}
}
return true;
}
}
Email/Sms 观察者
/**
* 观察者通过 update 来接受主题的更新通知
*/
class EmailObserver implements SplObserver
{
/**
* 观察者接收主题通知的接口
* @param SplSubject $user [description]
* @return [type] [description]
*/
public function update(SplSubject $user)
{
echo "send email to " . $user->email . PHP_EOL;
}
}
class SmsObserver implements SplObserver
{
public function update(SplSubject $user)
{
echo "send sms to " . $user->mobile . PHP_EOL;
}
}
业务
// User 主题
$user = new User();
// 为 user 注册 Email 观察者 (Email 观察者订阅 User 主题)
$emailObserver = new EmailObserver();
$user->attach($emailObserver);
// 为 user 注册 Sms 观察者 (Sms 观察者订阅 User 主题)
$smsObserver = new SmsObserver();
$user->attach($smsObserver);
// 从 user 上删除 Sms 观察者 (Sms 观察者取消订阅 User 主题)
//$user->detach($smsObserver);
// register 中会根据注册结果通知观察者 观察者做相应的处理
$user->register("big cat", "32448732@qq.com", "1888888888");
结果
send email to 32448732@qq.com
send sms to 1888888888
[Finished in 0.1s]
其实观察者模式类似于事件注册和钩子回调,日常开发中我们可能重构分离出一部分类的行为到外部,封装成独立的功能模块,在注册到类中,可以使用事件注册,也可以使用观察者模式
345374113@qq.com 贡献 FUNC
版
/** *******观察者模式FUNC版*******/
$user = [
'name' => '',
'email' => '',
'mobile' => '',
'updates' => [],
'OnUpdate' => function ($update) use (&$user) {
array_push($user ['updates'],$update);
},
'Update' => function() use (&$user ) {
foreach ($user ['updates'] as $update) $update($user);
},
'Register' => function($name, $email, $mobile) use (&$user) {
$user['name'] = $name;
$user['email'] = $email;
$user['mobile'] = $mobile;
$user['Update']();
},
];
$user["OnUpdate"](function ($subject){
echo "send email to " . $subject['email'] . PHP_EOL;
});
$user["OnUpdate"](function ($subject){
echo "send sms to " . $subject['mobile'] . PHP_EOL;
});
$user['Register']("紫羽·大薯", "345374113@qq.com", "1888888888");
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。